- 
                Notifications
    You must be signed in to change notification settings 
- Fork 8k
Fix error handling during opcache compilation #18541
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Conversation
…ile()` The logic to disable userland error handlers existed since the first commit including OPcache in php-src. The reason unfortunately is not explained. No existing tests require that userland error handlers do not trigger in `opcache_compile_file()`. The newly added tests are intended to exercise all possible kinds of edge-cases that might arise and cause issues (e.g. throwing Exceptions, performing a bailout, or loading a different file from within the error handler). They all pass for both `--repeat 1` and `--repeat 2`, as well as with OPcache enabled and without OPcache enabled (in the latter case with the exception of the `opcache_get_status()` verification). The list of cached files at the end of execution also matches my expectations. Therefore it seems that disabling the userland error handler when compiling a file with OPcache is not (or at least no longer) necessary. Fixes php#17422.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Some first remarks regarding the tests. Did not yet work through the C code.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Should we also have a test that tries to catch the exception or are the new tests in ext/opcache/tests/gh17422/ sufficient?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This one fails when OPcache is not loaded. It still says the “During inheritance of C” bit. Thus:
We could apply the same behavior when opcache is disabled.
This would make sense to me.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
The changes all make sense to me as far as I'm qualified to tell 😄
| A thought before reviewing this in detail: Should this behavior be unified for opcache / no opcache? Consider this example: // a.php
set_error_handler(function () {
    require __DIR__ . '/c.php';
    Foo::test();
});
require __DIR__ . '/b.php';
// b.php
class Foo  {
    public static function test() {
        switch (true) {
            default:
                continue;
        }
        echo "b.php\n";
    }
}
// c.php
class Foo  {
    public static function test() {
        echo "c.php\n";
    }
}This behavior will diverge currently, printing b.php with opcache, but  | 
| 
 @iluuu1994 From the initial PR description. And I agree that unifying the behavior makes sense. The issue only existed in the first place, because the behavior was not consistent between OPcache and non-OPcache. To me any observable behavioral differences introduced by OPcache are a bug (unless they are the main purpose of OPcache, e.g. autoloading seeing outdates files when revalidation is disabled). | 
| Ah thank you, I missed this last sentence. 🙂 And Arnaud is fast at implementing than me commenting. 😄 
 I think diverging behavior, especially in incorrect code, is ok if the engine benefits from it. As an example, cache slot merging is observable, but it's a worthwhile trade-off. | 
… generated tmp-php.ini
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I think, if we like to fix the inconsistency problem once and forever we should use the same behaviour with and without opcache. So we should always delay error messages and user error handlers.
I don't object against this PR, but I can't think about all possible consequences now.
| Thank you everyone for the comments and remarks. With the latest changes, the behavior is almost the same when opcache is enabled or disabled: 
 One remaining notable difference is that when  | 
| 
 We could consider using a different default for  | 
| Could this be considered as a bugfix? | 
| 
 To me this seems too dangerous and complex for a stable branch. | 
| The thing is this is pretty bad on the community side, with deprecations that cannot be handled the proper way. | 
| Or maybe we can just not trigger this deprecation until 8.5? | 
| 
 It's not up to me, but if I were to have to make a choice, then this would be my preference. | 
This PR was merged into the 6.4 branch. Discussion ---------- Silence E_DEPRECATED and E_USER_DEPRECATED | Q | A | ------------- | --- | Branch? | 6.4 | Bug fix? | yes | New feature? | no | Deprecations? | no | Issues | Fix #60528 | License | MIT This change works around php/php-src#17422 This shouldn't change much in practice because we report both not-silenced and silenced deprecations, except for the case when the native php error handler is triggered, which is what happens at this opcache stage, see php/php-src#18541 also. Commits ------- f21b2f4 Silence E_DEPRECATED and E_USER_DEPRECATED
This PR was merged into the 6.4 branch. Discussion ---------- Silence E_DEPRECATED and E_USER_DEPRECATED | Q | A | ------------- | --- | Branch? | 6.4 | Bug fix? | yes | New feature? | no | Deprecations? | no | Issues | Fix #60528 | License | MIT This change works around php/php-src#17422 This shouldn't change much in practice because we report both not-silenced and silenced deprecations, except for the case when the native php error handler is triggered, which is what happens at this opcache stage, see php/php-src#18541 also. Commits ------- f21b2f4df21 Silence E_DEPRECATED and E_USER_DEPRECATED
| I'm going to merge this. @nicolas-grekas I'm happy making the change to not trigger this deprecation in older versions, but this probably requires a email to internals. | 
| On Symfony's side we've dealt with this by silencing deprecations by default using error_level() That means things are under control in this part of the ecosystem. Dunno how other parts are affected. | 
| Thanks for fixing this BTW! | 
Introduced by phpGH-18541. This path is hit when compilation fails with a compile error, rather than bailout. If a non-fatal error is recorded, it will not be emitted nor freed. Handle accordingly.
Introduced by GH-18541. This path is hit when compilation fails with a compile error, rather than bailout. If a non-fatal error is recorded, it will not be emitted nor freed. Handle accordingly.
Possible fix for GH-17422
Based on #17627
Current behavior
Currently, Opcache disables the user-defined error handler during compilation. When
opcache.record_errorsis enabled, errors are replayed when loading the script from cache later, this time without disabling the user-defined error handler.Changes
This PR makes the following changes:
Errors emitted during compilation are delayed and handled after compilation, when it's safe to execute user-defined error handlers. As before, when
opcache.record_errorsis enabled, errors are also replayed when loading the script from cache and user-defined error handler are called, if any.This is done by changing the
EG(record_errors)mechanism so that errors are not only recorded, but also delayed.Class linking:
Since class linking uses the same mechanism, errors emitted during class linking are also delayed (with Opcache), and replayed after linking. Exceptions thrown by user-defined error handlers are not promoted to fatal errors anymore (when opcache is enabled).
Fatal errors:
Fatal errors are not safe to delay, so they are always processed immediately. Any delayed errors are also processed, but unfortunately it's not safe to execute user-defined error handlers at this point, so these errors still bypass them.
Notes
This targets master because of the behavior changes.
We could apply the same behavior when opcache is disabled. (Edit: done)
UPGRADING draft: